<!DOCTYPE html>
<html lang="zh-cn">
<head>
  <meta charset="utf-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
  <title>使用SNTP协议从时间服务器同步时间 - whowin - 发表我个人原创作品的技术博客</title>
  <meta name="renderer" content="webkit" />
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>

<meta http-equiv="Cache-Control" content="no-transform" />
<meta http-equiv="Cache-Control" content="no-siteapp" />

<meta name="theme-color" content="#f8f5ec" />
<meta name="msapplication-navbutton-color" content="#f8f5ec">
<meta name="apple-mobile-web-app-capable" content="yes">
<meta name="apple-mobile-web-app-status-bar-style" content="#f8f5ec">


<meta name="author" content="whowin" /><meta name="description" content="在互联网上校准时间，是几乎连接在互联网上的每台计算机都要去做的事情，而且很多是在后台完成的，并不需要人工干预；互联网上有很多时间服务器可以发布精确的时间，计算机客户端使用NTP(Network Time Protocol)协议与这些时间服务器进行时间同步，使本机得到精确时间，本文简要描述了NTP协议的原理，对NTP协议的时间同步精度做了简要分析，并具体实现了SNTP(Simple Network Time Protocol)下的客户端，本文附有完整的C语言SNTP客户端的源程序。阅读本文只需掌握基本的socket编程即可，本文对网络编程的初学者难度不大。
" /><meta name="keywords" content="linux, socket, hugo, dos" />






<meta name="generator" content="Hugo 0.97.3 with theme even" />


<link rel="canonical" href="https://whowin.gitee.io/post/blog/network/0017-sync-time-from-time-server-using-sntp/" />
<link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png">
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png">
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png">
<link rel="manifest" href="/manifest.json">
<link rel="mask-icon" href="/safari-pinned-tab.svg" color="#5bbad5">



<link href="/sass/main.min.e3fea119b1980e848b03dffbeddb11dd0fba483eed0e5f11870fb8e31f145bbd.css" rel="stylesheet">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fancyapps/fancybox@3.1.20/dist/jquery.fancybox.min.css" integrity="sha256-7TyXnr2YU040zfSP+rEcz29ggW4j56/ujTPwjMzyqFY=" crossorigin="anonymous">


<meta property="og:title" content="使用SNTP协议从时间服务器同步时间" />
<meta property="og:description" content="在互联网上校准时间，是几乎连接在互联网上的每台计算机都要去做的事情，而且很多是在后台完成的，并不需要人工干预；互联网上有很多时间服务器可以发布精确的时间，计算机客户端使用NTP(Network Time Protocol)协议与这些时间服务器进行时间同步，使本机得到精确时间，本文简要描述了NTP协议的原理，对NTP协议的时间同步精度做了简要分析，并具体实现了SNTP(Simple Network Time Protocol)下的客户端，本文附有完整的C语言SNTP客户端的源程序。阅读本文只需掌握基本的socket编程即可，本文对网络编程的初学者难度不大。" />
<meta property="og:type" content="article" />
<meta property="og:url" content="https://whowin.gitee.io/post/blog/network/0017-sync-time-from-time-server-using-sntp/" /><meta property="article:section" content="post" />
<meta property="article:published_time" content="2023-02-13T16:43:29+08:00" />
<meta property="article:modified_time" content="2023-02-13T16:43:29+08:00" />

<meta itemprop="name" content="使用SNTP协议从时间服务器同步时间">
<meta itemprop="description" content="在互联网上校准时间，是几乎连接在互联网上的每台计算机都要去做的事情，而且很多是在后台完成的，并不需要人工干预；互联网上有很多时间服务器可以发布精确的时间，计算机客户端使用NTP(Network Time Protocol)协议与这些时间服务器进行时间同步，使本机得到精确时间，本文简要描述了NTP协议的原理，对NTP协议的时间同步精度做了简要分析，并具体实现了SNTP(Simple Network Time Protocol)下的客户端，本文附有完整的C语言SNTP客户端的源程序。阅读本文只需掌握基本的socket编程即可，本文对网络编程的初学者难度不大。"><meta itemprop="datePublished" content="2023-02-13T16:43:29+08:00" />
<meta itemprop="dateModified" content="2023-02-13T16:43:29+08:00" />
<meta itemprop="wordCount" content="6080">
<meta itemprop="keywords" content="Linux,网络编程,NTP,SNTP," /><meta name="twitter:card" content="summary"/>
<meta name="twitter:title" content="使用SNTP协议从时间服务器同步时间"/>
<meta name="twitter:description" content="在互联网上校准时间，是几乎连接在互联网上的每台计算机都要去做的事情，而且很多是在后台完成的，并不需要人工干预；互联网上有很多时间服务器可以发布精确的时间，计算机客户端使用NTP(Network Time Protocol)协议与这些时间服务器进行时间同步，使本机得到精确时间，本文简要描述了NTP协议的原理，对NTP协议的时间同步精度做了简要分析，并具体实现了SNTP(Simple Network Time Protocol)下的客户端，本文附有完整的C语言SNTP客户端的源程序。阅读本文只需掌握基本的socket编程即可，本文对网络编程的初学者难度不大。"/>

<!--[if lte IE 9]>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/classlist/1.1.20170427/classList.min.js"></script>
<![endif]-->

<!--[if lt IE 9]>
  <script src="https://cdn.jsdelivr.net/npm/html5shiv@3.7.3/dist/html5shiv.min.js"></script>
  <script src="https://cdn.jsdelivr.net/npm/respond.js@1.4.2/dest/respond.min.js"></script>
<![endif]-->

  <script async src="/js/busuanzi.pure.mini.js"></script><script async src="https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js?client=ca-pub-9724909319263152"
     crossorigin="anonymous"></script>


</head>
<body>
  <div id="mobile-navbar" class="mobile-navbar">
  <div class="mobile-header-logo">
    <a href="/" class="logo">WhoWin</a>
  </div>
  <div class="mobile-navbar-icon">
    <span></span>
    <span></span>
    <span></span>
  </div>
</div>
<nav id="mobile-menu" class="mobile-menu slideout-menu">
  <ul class="mobile-menu-list">
    <a href="/">
        <li class="mobile-menu-item">首页</li>
      </a><a href="/post/">
        <li class="mobile-menu-item">文章归档</li>
      </a><a href="/article-categories/categories/">
        <li class="mobile-menu-item">文章分类</li>
      </a><a href="/tags/">
        <li class="mobile-menu-item">文章标签</li>
      </a><a href="/about/about/">
        <li class="mobile-menu-item">关于</li>
      </a>
  </ul>

  


</nav>

  <div class="container" id="mobile-panel">
    <header id="header" class="header">
        <div class="logo-wrapper">
  <a href="/" class="logo">WhoWin</a>
  
  <div style="position:absolute; left: 80px; top: 75px; color: crimson">
      ———开源和分享是技术发展的源泉和动力；本博客所有文章均为原创
  </div>
</div>





<nav class="site-navbar">
  <ul id="menu" class="menu">
    <li class="menu-item">
        <a class="menu-item-link" href="/">首页</a>
      </li><li class="menu-item">
        <a class="menu-item-link" href="/post/">文章归档</a>
      </li><li class="menu-item">
        <a class="menu-item-link" href="/article-categories/categories/">文章分类</a>
      </li><li class="menu-item">
        <a class="menu-item-link" href="/tags/">文章标签</a>
      </li><li class="menu-item">
        <a class="menu-item-link" href="/about/about/">关于</a>
      </li>
  </ul>
</nav>

    </header>

    <main id="main" class="main">
      <div class="content-wrapper">
        <div id="content" class="content">
          <article class="post">
    
    <header class="post-header">
      <h1 class="post-title">使用SNTP协议从时间服务器同步时间</h1>

      <div class="post-meta">
        <span class="post-time"> 2023-02-13 </span>
        <div class="post-category">
            <a href="/categories/linux/"> Linux </a>
            <a href="/categories/c-language/"> C Language </a>
            <a href="/categories/network/"> Network </a>
            </div>
        
      </div>
    </header>

    <div class="post-toc" id="post-toc">
  <h2 class="post-toc-title">文章目录</h2>
  <div class="post-toc-content always-active">
    <nav id="TableOfContents">
  <ul>
    <li>
      <ul>
        <li><a href="#1-ntp协议和sntp协议">1. NTP协议和SNTP协议</a></li>
        <li><a href="#2-ntp时间同步的基本原理">2. NTP时间同步的基本原理</a></li>
        <li><a href="#3-ntp和sntp协议参考">3. NTP和SNTP协议参考</a></li>
        <li><a href="#4-sntpntp协议的消息结构">4. SNTP(NTP)协议的消息结构</a></li>
        <li><a href="#5-sntp时间同步的精度分析">5. SNTP时间同步的精度分析</a></li>
        <li><a href="#4sntp客户端实例">4、SNTP客户端实例</a></li>
        <li><a href="#欢迎订阅-网络编程专栏httpsblogcsdnnetwhowincategory_12180345html"><strong>欢迎订阅 <a href="https://blog.csdn.net/whowin/category_12180345.html">『网络编程专栏』</a></strong></a></li>
      </ul>
    </li>
  </ul>
</nav>
  </div>
</div>
    <div class="post-content">
      <p>在互联网上校准时间，是几乎连接在互联网上的每台计算机都要去做的事情，而且很多是在后台完成的，并不需要人工干预；互联网上有很多时间服务器可以发布精确的时间，计算机客户端使用NTP(Network Time Protocol)协议与这些时间服务器进行时间同步，使本机得到精确时间，本文简要描述了NTP协议的原理，对NTP协议的时间同步精度做了简要分析，并具体实现了SNTP(Simple Network Time Protocol)下的客户端，本文附有完整的C语言SNTP客户端的源程序。阅读本文只需掌握基本的socket编程即可，本文对网络编程的初学者难度不大。</p>
<h2 id="1-ntp协议和sntp协议">1. NTP协议和SNTP协议</h2>
<ul>
<li>SNTP协议使用与NTP协议同样的报文结构和格式，所以仅就从服务器进行时间同步而言，在服务器端看NTP和SNTP没有什么区别，使用SNTP协议的客户端可以从任何一台符合NTP协议的时间服务器上进行时间同步；</li>
<li>NTP和SNTP协议的区别在于错误检测和时间校准的算法上，这主要体现在客户端的软件上；</li>
<li>SNTP客户端程序向一台NTP时间服务器发出时间数据包，接收来自服务器的回应，并据此计算本机的时间偏差，从而校准本机时间；</li>
<li>NTP协议的算法比SNTP复杂得多，NTP通常使用多个时间服务器校验时间，该算法使用多种方法来确定这些获取的时间值是否准确，包括模糊因子和识别与其他时间服务器不一致的时间服务器，然后加速或减慢系统时钟的<strong>漂移率</strong>，使系统时间可以做到
<ol>
<li>系统的时间总是正确的；</li>
<li>在初始校正时间后，系统时间不会再有任何时间跳跃。</li>
</ol>
</li>
<li>与NTP客户端不同，SNTP客户端通常只使用一个时间服务器来计算时间，然后将系统时间跳转到计算的时间；为了防止时间服务器出现不可用的情况，SNTP客户端可以有一个或多个备份时间服务器，但不会同时使用多个时间服务器来计算时间；</li>
<li>SNTP客户端通常按照一个固定的间隔时间去访问时间服务器，在间隔期间则不对系统时间做任何调整，所以，当SNTP访问时间服务器校准时间时，往往会产生时间跳跃；</li>
<li>我们可以用一个更形象的例子来说明SNTP客户端，我们把墙上的挂钟当做时间服务器，把我们戴的手表当做客户端
<ul>
<li>当我们的手表为SNTP客户端时，我们每隔一小时看一下挂钟，并使用挂钟来校准手表；</li>
<li>当手表12:00时，挂钟的时间为11:59:57秒，手表快了3秒钟，所以把手表调慢3秒钟；</li>
<li>在接下来的1个小时里，不会对手表做任何调整，当手表1:00时，挂钟的时间为12:59:57秒，所以我们再次把手表调慢3秒钟；</li>
<li>从手表时间的准确度来说，刚刚校准完时间时是最准确的，然后准确度逐渐变差，在再次调整时间前，其准确度是最差的；</li>
</ul>
</li>
<li>如果这样的情况能够满足你对时间的要求，那就可以使用SNTP协议去校准时间，否则就要考虑使用NTP客户端；</li>
<li>NTP客户端计算&quot;手表&quot;的时间变化方向和速率，以此为基础来补偿&quot;手表&quot;上的时间漂移，对&quot;手表&quot;进行实时调整，使&quot;手表&quot;一直保持准确；</li>
<li>实际上，对大多数PC而言，SNTP都是可以满足要求的，windows的内建程序w32time采用的就是SNTP协议。</li>
</ul>
<h2 id="2-ntp时间同步的基本原理">2. NTP时间同步的基本原理</h2>
<ul>
<li>
<p>NTP的原理是通过一个时间消息包的传送计算出客户端和服务器端的时间偏差，从而校准客户端的时间；</p>
</li>
<li>
<p>NTP和SNTP客户端均使用UDP向时间服务器发送消息，IANA为NTP分配的端口号为123，也就是说，NTP/SNTP客户端需要向时间服务器的123端口发送一个符合格式(下一节介绍消息格式)的UDP消息，客户端的接收端口号没有规定；</p>
</li>
<li>
<p>时间服务器是Server，客户端是Client，同步过程如下进行：</p>
<ol>
<li>Client向Server发送一个消息包，记录发出消息包时的时间戳 <strong>T<sub>1</sub></strong>(以Client系统时间为准)</li>
<li>Server收到消息包立即记录时间戳 <strong>T<sub>2</sub></strong>(以Server系统时间为准)</li>
<li>Server向Client返回一个消息包，返回消息包时记录时间戳 <strong>T<sub>3</sub></strong>(以Server系统时间为准)</li>
<li>Client收到Server返回的消息包，此时记录时间戳 <strong>T<sub>4</sub></strong>(以Client系统时间为基准</li>
</ol>
</li>
<li>
<p>过程如下图所示：</p>
<p><img src="https://whowin.gitee.io/images/180017/process-of-time-synchronization.png" alt="Time Synchronization"></p>
</li>
<li>
<p>T<sub>4</sub> 和 T<sub>1</sub> 是以Client的时间标准记录的时间戳，其差 T<sub>4</sub> - T<sub>1</sub> 表示整个消息传递过程所花费的总时间；</p>
</li>
<li>
<p>T<sub>3</sub> 和 T<sub>2</sub> 是以Server的时间标准记录的时间戳，其差 T<sub>3</sub> - T<sub>2</sub> 表明消息传递过程在Server停留的时间；</p>
</li>
<li>
<p>那么 (T<sub>4</sub> - T<sub>1</sub>) - (T<sub>3</sub> - T<sub>2</sub>) 应该就是信息包的往返时间(总时间-在Server停留的时间)；</p>
</li>
<li>
<p><strong>如果假定信息包从Client到Server和从Server到Client所用的时间一样</strong>，那么，从Client到Server或者从Server到Client信息包的传送时间d为：
$$
\large \ \ d = {(T_4 - T_1) - (T_3 - T_2) \over 2}
$$</p>
</li>
<li>
<p>假定Client相对于Server机的时间误差是 <strong>t</strong>(t = 服务器时间戳 - 客户端时间戳)，则有下列等式：
$$
\begin{cases}
T_2 = T_1 + t + d \newline
T_4 = T_3 - t + d
\end{cases}
$$</p>
</li>
<li>
<p>从以上三个等式组成一个方程式：
$$
\begin{cases}
\large d = {(T_4 - T_1) - (T_3 - T_2) \over 2} \newline
\normalsize T_2 = T_1 + t + d \newline
T_4 = T_3 - t + d
\end{cases}
$$</p>
</li>
<li>
<p>可以解出Client机的时间误差 <strong>t</strong> 为：
$$
\large t = {(T_2 - T_1) + (T_3 - T_4) \over 2}
$$</p>
</li>
<li>
<p>如果一时没有转过来，可以自己在纸上画个图，再细细地琢磨一下，应该没有问题。</p>
</li>
</ul>
<h2 id="3-ntp和sntp协议参考">3. NTP和SNTP协议参考</h2>
<ul>
<li>这里列出这两个协议的原件下载地址，有兴趣的读者可以认真读一下；</li>
<li>NTP最新版目前是NTP v4(RFC5905)，v5还没有正是推出，这里列出NTP v4和v3(RFC1305)的链接，还有一个NTP v3的pdf文档的下载地址；</li>
<li>SNTP目前的版本是v4(RFC4330)，其实这个版本也是很久以前的，2006年的，SNTP一直没有新的版本。</li>
<li><a href="https://www.rfc-editor.org/rfc/rfc5905">Network Time Protocol Version 4</a></li>
<li><a href="https://www.rfc-editor.org/rfc/rfc1305">Network Time Protocol (Version 3)</a></li>
<li><a href="https://www.rfc-editor.org/rfc/rfc1305.pdf">Network Time Protocol (Version 3) pdf</a></li>
<li><a href="https://www.rfc-editor.org/rfc/rfc4330">Simple Network Time Protocol (SNTP) Version 4</a></li>
</ul>
<h2 id="4-sntpntp协议的消息结构">4. SNTP(NTP)协议的消息结构</h2>
<ul>
<li>
<p>下面这张图取自SNTP协议，很直观地显示出NTP消息包的结构，要注意的是，所有字段应该都是网络字节序(big endian)</p>
</li>
<li>
<p>NTP时间同步过程中，客户端发送的消息包结构与时间服务器返回的消息包结构是完全一样的，时间服务器会将其中的一些字段改写，然后发回；</p>
<!--![NTP packet struct][img03]-->
<div class="highlight"><div class="chroma">
<table class="lntable"><tr><td class="lntd">
<pre tabindex="0" class="chroma"><code><span class="lnt"> 1
</span><span class="lnt"> 2
</span><span class="lnt"> 3
</span><span class="lnt"> 4
</span><span class="lnt"> 5
</span><span class="lnt"> 6
</span><span class="lnt"> 7
</span><span class="lnt"> 8
</span><span class="lnt"> 9
</span><span class="lnt">10
</span><span class="lnt">11
</span><span class="lnt">12
</span><span class="lnt">13
</span><span class="lnt">14
</span><span class="lnt">15
</span><span class="lnt">16
</span><span class="lnt">17
</span><span class="lnt">18
</span><span class="lnt">19
</span><span class="lnt">20
</span><span class="lnt">21
</span><span class="lnt">22
</span><span class="lnt">23
</span><span class="lnt">24
</span><span class="lnt">25
</span><span class="lnt">26
</span><span class="lnt">27
</span><span class="lnt">28
</span><span class="lnt">29
</span><span class="lnt">30
</span><span class="lnt">31
</span><span class="lnt">32
</span><span class="lnt">33
</span><span class="lnt">34
</span><span class="lnt">35
</span></code></pre></td>
<td class="lntd">
<pre tabindex="0" class="chroma"><code class="language-plaintext" data-lang="plaintext"><span class="line"><span class="cl">                     1                   2                   3
</span></span><span class="line"><span class="cl"> 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|LI | VN  |Mode |    Stratum    |     Poll      |   Precision   |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                          Root Delay                           |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                       Root Dispersion                         |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                     Reference Identifier                      |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                   Reference Timestamp (64)                    |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                   Originate Timestamp (64)                    |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                    Receive Timestamp (64)                     |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                    Transmit Timestamp (64)                    |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                 Key Identifier (optional) (32)                |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                 Message Digest (optional) (128)               |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">|                                                               |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span></code></pre></td></tr></table>
</div>
</div></li>
<li>
<p><strong>LI(Leap Indicator)</strong></p>
<ul>
<li>闰秒指示，2 bits，bit 0和bit 1</li>
<li>表示是否警告在当天的最后一分钟插入/删除一个闰秒；通常填0表示不需要警告。</li>
<li>这个值只在服务器端有意义，SNTP协议第4节中对LI可能的取值做了说明</li>
</ul>
</li>
<li>
<p><strong>VN(Version Number)</strong></p>
<ul>
<li>NTP/SNTP版本号，3 bits</li>
<li>这里只要填一个服务器支持的版本即可，目前最高的版本为4，填4，估计填3也不会有问题，因为绝大多数的时间服务器应该仍然支持NTP v3，以保证一些古老的客户端仍可以运行；</li>
</ul>
</li>
<li>
<p><strong>Mode</strong></p>
<ul>
<li>工作模式，3 bits，其取值含义如下
<div class="highlight"><div class="chroma">
<table class="lntable"><tr><td class="lntd">
<pre tabindex="0" class="chroma"><code><span class="lnt"> 1
</span><span class="lnt"> 2
</span><span class="lnt"> 3
</span><span class="lnt"> 4
</span><span class="lnt"> 5
</span><span class="lnt"> 6
</span><span class="lnt"> 7
</span><span class="lnt"> 8
</span><span class="lnt"> 9
</span><span class="lnt">10
</span></code></pre></td>
<td class="lntd">
<pre tabindex="0" class="chroma"><code class="language-plaintext" data-lang="plaintext"><span class="line"><span class="cl">Mode  Meaning
</span></span><span class="line"><span class="cl">------------------------------------
</span></span><span class="line"><span class="cl"> 0    reserved
</span></span><span class="line"><span class="cl"> 1    symmetric active
</span></span><span class="line"><span class="cl"> 2    symmetric passive
</span></span><span class="line"><span class="cl"> 3    client
</span></span><span class="line"><span class="cl"> 4    server
</span></span><span class="line"><span class="cl"> 5    broadcast
</span></span><span class="line"><span class="cl"> 6    reserved for NTP control message
</span></span><span class="line"><span class="cl"> 7    reserved for private use
</span></span></code></pre></td></tr></table>
</div>
</div></li>
<li>当访问单台时间服务器同步时间时，仅取值3和4有意义，客户端发送消息时将这个字段填3，表示发送消息者是一个客户端；服务器回复消息时将这个字段改为4，表示回复消息的是服务器；</li>
</ul>
</li>
<li>
<p><strong>Stratum</strong></p>
<ul>
<li>表示当前时间服务器在NTP网络体系中所处的层，是一个8位无符号整数，其值通常为0-15；该字段仅在NTP服务器消息中有意义；</li>
<li>NTP 的网络体系是分层(stratum)结构，Stratum-0层设备(包括原子钟和gps钟)是最精确的，但不能通过网络向客户端授时；0级设备通常仅作为1级时间服务器的参考时钟(或同步源)；</li>
<li>Stratum-1层设备是可以通过网络授时的最准确的ntp时间源，1层设备通常通过0层参考时钟同步时间；</li>
<li>Stratum-2层设备通过网络连接从一级设备同步时间；由于网络抖动和延迟，二级服务器的时间准确度不如一级服务器；从第二层时间源同步的NTP客户端将是Stratum-3设备；</li>
<li>以此类推，层级越高，其时间的精确度和可靠度越低；</li>
<li>NTP协议不允许客户端接受来自Stratum-15以上设备的时间，因此Stratum-15是最低的NTP层；</li>
</ul>
</li>
<li>
<p><strong>Poll Interval</strong></p>
<ul>
<li>表示连续时间消息之间的最大间隔，以2的指数表示(比如4则间隔时间为 2<sup>4</sup>)，单位为秒，此值为一个8位无符号整数；</li>
<li>该字段仅在SNTP服务器消息中有意义，其值范围为4(16秒)到17(131,072秒——大约36小时)；</li>
</ul>
</li>
<li>
<p><strong>Precision</strong></p>
<ul>
<li>时间服务器的系统时钟精度，以2的指数表示(比如-10则精度 2<sup>-10</sup>)，单位为秒，此值为一个8位有符号整数；</li>
<li>此字段仅在服务器消息中有意义，其中的值范围从-6(主频时钟)到-20(某些工作站中的微秒时钟)；</li>
</ul>
</li>
<li>
<p><strong>Root Delay</strong></p>
<ul>
<li>表示时间服务器到主参考源的总往返延迟，以秒为单位，是一个32位有符号的定点数，小数点在第bit 15和bit 16之间；</li>
<li>该变量可以取正值，也可以取负值，取决于相对时间和频率偏移量，该字段仅在服务器消息中有意义，其值范围从几毫秒的负值到几百毫秒的正值；</li>
</ul>
</li>
<li>
<p><strong>Root Dispersion</strong></p>
<ul>
<li>表示相对于主要时钟参考源的的最大误差，单位为秒，是一个32位有符号的定点数，小数点在第bit 15和bit 16之间；</li>
<li>其值只能是大于0的正数；</li>
</ul>
</li>
<li>
<p><strong>Reference Identifier</strong></p>
<ul>
<li>用于标识特定的时间参考源，32位串；此字段仅在服务器消息中有意义；</li>
<li>对于Stratum-0和Statum-1，该字段的值为一个四字节的ASCII字符串，左对齐并以0填充到32位；</li>
<li>对于IPv4的Stratum-2时间服务器，该字段的值为时间同步源的32位IPv4地址(IP地址)；</li>
</ul>
</li>
<li>
<p><strong>Reference Timestamp</strong></p>
<ul>
<li>本地时钟最后一次设置或修正时的时间，64位时间戳格式。</li>
</ul>
</li>
<li>
<p><strong>Originate Timestamp</strong></p>
<ul>
<li>前面原理部分说到的 T<sub>1</sub>，也就是消息包从客户端发出时，客户端系统时间戳，由客户端程序填写，64位时间戳格式；</li>
</ul>
</li>
<li>
<p><strong>Receive Timestamp</strong></p>
<ul>
<li>前面原理部分说到的 T<sub>2</sub>，也就是服务器收到客户端消息包时，服务器端系统时间戳，由服务器端程序填写，64位时间戳格式；</li>
</ul>
</li>
<li>
<p><strong>Transmit Timestamp</strong></p>
<ul>
<li>前面原理部分说到的 T<sub>3</sub>，也就是服务器把数据包返回客户端时，服务器端系统时间戳，由服务器端程序填写，64位时间戳格式；</li>
</ul>
</li>
<li>
<p><strong>Authenticator</strong></p>
<ul>
<li>可选项，一般不填；</li>
<li>当采用NTP认证方案时，&ldquo;Key Identifier&quot;和&quot;Message Digest&quot;字段包含了<a href="https://www.rfc-editor.org/rfc/rfc1305">《RFC 1305》</a>附录C中定义的MAC(Message authentication code)信息。</li>
</ul>
</li>
<li>
<p>关于<strong>64位时间戳</strong></p>
<ul>
<li>
<p>前面多次提到<strong>64位时间戳</strong>，在NTP/SNTP协议中，对此有专门的定义</p>
</li>
<li>
<p>NTP时间戳表示为 64 位无符号定点数，以秒为单位，相对于1900年1月1日的0时；</p>
</li>
<li>
<p>整数部分在前32位，小数部分在后32位；在小数部分，没有意义的低位，通常要设置为0；</p>
</li>
<li>
<p>有关小数部分(Fraction Part)，其单位既不是毫秒(millisecond)，也不是微秒(microsecond)，<strong>其单位为 1/2<sup>32</sup> 秒</strong>，这一点非常重要，但却鲜有文章说明，如果不知道这一点，UNIX时间戳和NTP时间戳之间的转换就搞不明白；</p>
</li>
<li>
<p>下图摘自SNTP协议</p>
<!--![NTP timestamp structure][img04]-->
<div class="highlight"><div class="chroma">
<table class="lntable"><tr><td class="lntd">
<pre tabindex="0" class="chroma"><code><span class="lnt">1
</span><span class="lnt">2
</span><span class="lnt">3
</span><span class="lnt">4
</span><span class="lnt">5
</span><span class="lnt">6
</span><span class="lnt">7
</span></code></pre></td>
<td class="lntd">
<pre tabindex="0" class="chroma"><code class="language-plaintext" data-lang="plaintext"><span class="line"><span class="cl">                     1                   2                   3
</span></span><span class="line"><span class="cl"> 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                           Seconds                             |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span><span class="line"><span class="cl">|                  Seconds Fraction (0-padded)                  |
</span></span><span class="line"><span class="cl">+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
</span></span></code></pre></td></tr></table>
</div>
</div></li>
</ul>
</li>
</ul>
<h2 id="5-sntp时间同步的精度分析">5. SNTP时间同步的精度分析</h2>
<ul>
<li>
<p>有很多文章中都提到，NTP时间同步的精度可以达到50ms以内，本节将讨论这个精度是如何计算得出的；</p>
</li>
<li>
<p>要注意我们讲网络时间同步的精度并不包括时间服务器本身的时间精度，只是客户端与服务器端同步时间后，客户端与服务器端相对时间误差；</p>
</li>
<li>
<p>本文第2节介绍NTP时间同步的基本原理时，曾经列出过方程式，当时有一个重要的假设为：<strong>假定信息包从Client到Server和从Server到Client所用的时间一样</strong>，这次我们去掉这个假设，重新列出方程式；</p>
</li>
<li>
<p>假定消息包从Client到Server的时长为 d<sub>1</sub>，从Server到Client的时长为 d<sub>2</sub>，d<sub>2</sub> 与 d<sub>1</sub> 的差为  δ，其它定义同前，则有下列等式：
$$
\begin{cases}
\delta = d_2 - d_1 \newline
T_2 = T_1 + t + d_1 \newline
T_4 = T_3 - t + d_2 \newline
\end{cases}
$$</p>
</li>
<li>
<p>(将1式带入3式)
$$
\begin{cases}
t = T_2 - T_1 - d_1 \newline
t = T_3 - T_4 + \delta + d_1 \newline
\end{cases}
$$</p>
</li>
<li>
<p>两式相加，得出
$$
\large t = {{(T_2 - T_1) + (T_3 - T_4)} \over 2} + {\delta \over 2}
$$</p>
</li>
<li>
<p>这个结果是精确的，没有任何假设，理论时间同步的精度为 0；</p>
</li>
<li>
<p>当 δ 为 <strong>0</strong> 时，相当于 <strong>假定信息包从Client到Server和从Server到Client所用的时间一样</strong>，得出的结果和第 2 节的结果是一样的；</p>
</li>
<li>
<p>由此可见，误差由 δ 产生，而 δ 的最大值为 d<sub>2</sub> 或者最小值为 -d<sub>1</sub>，假定Client到Server的最大时延为100ms，则 δ 的最大值为 100ms，则根据上式，其时间精度的最大误差为 δ/2，即 50ms</p>
</li>
<li>
<p>由上面的计算可以得知，NTP协议进行时间同步的精度误差主要来自数据包从Client到Server和从Server到Client的时间不一样，这个差异越大，其误差越大；</p>
</li>
<li>
<p>SNTP使用UDP协议发送时间信息包，UDP又是一种无连接的协议，从Client到Server和从Server到Client的路由很可能是不一样的，这无形中会使时间同步的精度变差；</p>
</li>
<li>
<p>由上面的分析可以看出：<strong>找到一个时延比较小的时间服务器可以有效地提高时间同步的精度</strong>。</p>
</li>
</ul>
<h2 id="4sntp客户端实例">4、SNTP客户端实例</h2>
<ul>
<li>
<p>说起来一大堆，但实现起来其实并不像说的那么复杂。</p>
</li>
<li>
<p>SNTP协议允许使用单播(unicast)、广播(broadcast)和多播(manycast)模式，通常我们只能使用单播模式，广播和多播模式的时间服务器只存在于某个子网中，为有限的用户服务；互联网上并不存在实际的广播或多播模式的时间服务器；</p>
</li>
<li>
<p>根据<a href="https://www.rfc-editor.org/rfc/rfc5905">《SNTP 协议》</a>第5节&rdquo;<strong>SNTP Client Operations</strong>&ldquo;的说明，使用单播模式进行时间同步时，向时间服务器发送的请求数据包中，除了第一个字节以外，其他字段都可以设为0，也可以将Originate Timestamp(T<sub>1</sub>)填在Transmit Timestamp(T<sub>4</sub>)这个字段上，时间服务器会将Transmit Timestamp(T<sub>4</sub>)字段的内容搬移到Originate Timestamp(T<sub>1</sub>)上，然后填上正确的Transmit Timestamp(T<sub>4</sub>)；</p>
</li>
<li>
<p>下面的例子中就是按照SNTP协议的说法去做的，在发送的请求包中，填了LI、VN、MODE三个字段，并把T<sub>1</sub>(Originate Timestamp)填在了T<sub>4</sub>(Transmit Timestamp)上，从时间服务器返回的数据看，完全印证了SNTP协议中的说法；</p>
</li>
<li>
<p>要注意的是，ntp数据包中的各个字段都是网络字节序(big endian)，而我们使用的电脑都是主机字节序(little endian)，所以相互之间要做转换；</p>
</li>
<li>
<p>再次强调一下，NTP时间戳的fraction字段的单位是 <strong>1/2<sup>32</sup> 秒</strong>；</p>
</li>
<li>
<p>源程序文件名：<a href="https://gitee.com/whowin/whowin/blob/blog/sourcecodes/180017/sntp-client.c">sntp-client.c</a>(<strong>点击文件名下载源文件</strong>)</p>
</li>
<li>
<p>编译：<code>gcc -Wall sntp-client.c -o sntp-client -lm</code>，因为其中使用了数学函数，所以编译时要加上 &ldquo;-lm&rdquo;</p>
</li>
<li>
<p>运行：<code>./sntp-client ntp.aliyun.com</code></p>
</li>
<li>
<p>运行截图：</p>
<p><img src="https://whowin.gitee.io/images/180017/screenshot-of-sntp-client.png" alt="screenshot of sntp_client"></p>
</li>
<li>
<p>在源程序中，列出了几个时间服务器，可以通过百度或者谷歌找更多的时间服务器进行尝试。</p>
</li>
</ul>
<h2 id="欢迎订阅-网络编程专栏httpsblogcsdnnetwhowincategory_12180345html"><strong>欢迎订阅 <a href="https://blog.csdn.net/whowin/category_12180345.html">『网络编程专栏』</a></strong></h2>
<hr>
<p><strong>欢迎访问我的博客：https://whowin.cn</strong></p>
<p><strong>email: <a href="mailto:hengch@163.com">hengch@163.com</a></strong></p>
<p><img src="https://whowin.gitee.io/images/qrcode/sponsor-qrcode.png" alt="donation"></p>
<!-- CSDN
[img01]:https://img-blog.csdnimg.cn/img_convert/f2ce11d74bbd7545ab57d366ed662128.png
[img02]:https://img-blog.csdnimg.cn/img_convert/d3edbc42838b367eebd2bf8c026c4797.png
[img03]:https://img-blog.csdnimg.cn/img_convert/bf3ef6b56485c47857358e308a6068ee.png
[img04]:https://img-blog.csdnimg.cn/img_convert/b3f8951f77131c03474314fef8f8d515.png
-->
    </div>

    <div class="post-copyright">
  <p class="copyright-item">
    <span class="item-title">文章作者</span>
    <span class="item-content">whowin</span>
  </p>
  <p class="copyright-item">
    <span class="item-title">上次更新</span>
    <span class="item-content">
        2023-02-13
        
    </span>
  </p>
  
  
</div>
<footer class="post-footer">
      <div class="post-tags">
          <a href="/tags/linux/">Linux</a>
          <a href="/tags/%E7%BD%91%E7%BB%9C%E7%BC%96%E7%A8%8B/">网络编程</a>
          <a href="/tags/ntp/">NTP</a>
          <a href="/tags/sntp/">SNTP</a>
          </div>
      <nav class="post-nav">
        <a class="prev" href="/post/blog/network/0018-tun-example-for-setting-up-ip-tunnel/">
            <i class="iconfont icon-left"></i>
            <span class="prev-text nav-default">使用tun虚拟网络接口建立IP隧道的实例</span>
            <span class="prev-text nav-mobile">上一篇</span>
          </a>
        <a class="next" href="/post/blog/network/0016-longest-prefix-match/">
            <span class="next-text nav-default">简单的路由表查找程序</span>
            <span class="next-text nav-mobile">下一篇</span>
            <i class="iconfont icon-right"></i>
          </a>
      </nav>
    </footer>
  </article>
        </div>
        

  <span id="/post/blog/network/0017-sync-time-from-time-server-using-sntp/" class="leancloud_visitors" data-flag-title="使用SNTP协议从时间服务器同步时间">
		<span class="post-meta-item-text">文章阅读量 </span>
		<span class="leancloud-visitors-count">0</span>
		<p></p>
	  </span>
  <div id="vcomments"></div>
  <script src="//cdn1.lncld.net/static/js/3.0.4/av-min.js"></script>
  <script src='//unpkg.com/valine/dist/Valine.min.js'></script>
  <script type="text/javascript">
    new Valine({
        el: '#vcomments' ,
        appId: 'OFCGzCfJRUglzOdzrqMGkbTR-gzGzoHsz',
        appKey: 'v7P29kPAEbsmaavaYPNhGhnF',
        notify:  false ,
        verify:  false ,
        avatar:'mm',
        placeholder: '说点什么吧...',
        visitor:  true 
    });
  </script>

  

      </div>
    </main>

    <footer id="footer" class="footer">
      <div class="social-links">
      <a href="mailto:hengch@163.com" class="iconfont icon-email" title="email"></a>
  <a href="https://whowin.gitee.io/index.xml" type="application/rss+xml" class="iconfont icon-rss" title="rss"></a>
</div>
<div class="copyright">
  <span class="power-by">
    由 <a class="hexo-link" href="https://gohugo.io">Hugo</a> 强力驱动
  </span>
  <span class="division">|</span>
  <span class="theme-info">
    主题 - 
    <a class="theme-link" href="https://github.com/olOwOlo/hugo-theme-even">Even</a>
  </span>

  <div class="busuanzi-footer">
    <span id="busuanzi_container_site_pv"> 本站总访问量 <span id="busuanzi_value_site_pv"><img src="/img/spinner.svg" alt="spinner.svg"/></span> 次 </span>
      <span class="division">|</span>
    <span id="busuanzi_container_site_uv"> 本站总访客数 <span id="busuanzi_value_site_uv"><img src="/img/spinner.svg" alt="spinner.svg"/></span> 人 </span>
  </div>

  <span class="copyright-year">
    &copy; 
    2022 - 
    2024<span class="heart"><i class="iconfont icon-heart"></i></span><span>whowin</span>
  </span>
</div>

    </footer>

    <div class="back-to-top" id="back-to-top">
      <i class="iconfont icon-up"></i>
    </div>
  </div>
  
  <script src="https://cdn.jsdelivr.net/npm/jquery@3.2.1/dist/jquery.min.js" integrity="sha256-hwg4gsxgFZhOsEEamdOYGBf13FyQuiTwlAQgxVSNgt4=" crossorigin="anonymous"></script>
  <script src="https://cdn.jsdelivr.net/npm/slideout@1.0.1/dist/slideout.min.js" integrity="sha256-t+zJ/g8/KXIJMjSVQdnibt4dlaDxc9zXr/9oNPeWqdg=" crossorigin="anonymous"></script>
  <script src="https://cdn.jsdelivr.net/npm/@fancyapps/fancybox@3.1.20/dist/jquery.fancybox.min.js" integrity="sha256-XVLffZaxoWfGUEbdzuLi7pwaUJv1cecsQJQqGLe7axY=" crossorigin="anonymous"></script>



<script type="text/javascript" src="/js/main.min.64437849d125a2d603b3e71d6de5225d641a32d17168a58106e0b61852079683.js"></script>
  <script type="text/javascript">
    window.MathJax = {
      tex: {
        inlineMath: [['$','$'], ['\\(','\\)']],
        }
    };
  </script>
  <script async src="https://cdn.jsdelivr.net/npm/mathjax@3.0.5/es5/tex-mml-chtml.js" integrity="sha256-HGLuEfFcsUJGhvB8cQ8nr0gai9EucOOaIxFw7qxmd+w=" crossorigin="anonymous"></script>








</body>
</html>
